Simulation Experiment Recipe

Objectives

The objective of this simulation experiment is to provide a toy example on how to use simChef and showcase the automated R Markdown-generated documentation. For the sake of illustration, this toy simulation experiment studies the performance of linear regression at the surface-level with the sole purpose of facilitating an easy-to-understand walkthrough.

[Typically, the objective of the simulation experiment (and this blurb) will be more scientific than instructive and will warrant additional context/background and domain knowledge.]

Data Generation

Linear Gaussian DGP

In the Linear Gaussian DGP, we simulate the feature/design matrix \(\mathbf{X} \in \mathbb{R}^{n \times p}\) from a normal distribution and the response vector \(\mathbf{y} \in \mathbb{R}^n\) from a linear model. Specifically,

\[\begin{gather*}\mathbf{X} \sim N\left(\mathbf{0}, \begin{pmatrix} 1 & \rho \\ \rho & 1 \end{pmatrix}\right), \\\mathbf{y} = \mathbf{X} \boldsymbol{\beta} + \boldsymbol{\epsilon},\\\boldsymbol{\epsilon} \sim N(\mathbf{0}, \sigma^2 \mathbf{I}_n)\end{gather*}\]

Default Parameters in DGP

  • Number of samples: \(n = 200\)
  • Number of features: \(p = 2\)
  • Correlation among features: \(\rho = 0\)
  • Amount of noise: \(\sigma = 1\)
  • Coefficients: \(\boldsymbol{\beta} = (1, 0)^\top\)

[In practice, documentation of DGPs should answer the questions “what” and “why”. That is, “what” is the DGP, and “why” are we using/studying it? As this simulation experiment is a contrived example, we omit the “why” here.]

Function

#> function(n, beta, rho, sigma) {
#>   cov_mat <- matrix(c(1, rho, rho, 1), byrow = T, nrow = 2, ncol = 2)
#>   X <- MASS::mvrnorm(n = n, mu = rep(0, 2), Sigma = cov_mat)
#>   y <- X %*% beta + rnorm(n, sd = sigma)
#>   return(list(X = X, y = y))
#> }
#> <bytecode: 0x5603191d77f8>

Input Parameters

#> $n
#> [1] 200
#> 
#> $beta
#> [1] 1 0
#> 
#> $rho
#> [1] 0
#> 
#> $sigma
#> [1] 1

Methods and Evaluation

Methods

OLS

Given some data \(\mathbf{X}\) and \(\mathbf{y}\), we fit ordinary least squares (OLS) and examine the p-values for each coefficient in the model. The p-values are computed using a two-sided t-test (see summary.lm()).

Elaborating further on the testing, we are interested in testing:

\[\begin{align*}H_0: \beta_i = 0 \quad vs. \quad H_1: \beta_i \neq 0.\end{align*}\]

To test this, we compute the observed T-statistic, defined as

\[\begin{align*}T = \frac{\hat{\beta}_i}{\hat{SE}(\hat{\beta_i})},\end{align*}\]

and then compute the two-sided p-value under the t distribution with \(n - p - 1\) degrees of freedom. If the p-value is lower than some significance value \(\alpha\), then there is sufficient evidence to reject the null hypothesis \(H_0\). Otherwise, there is not sufficient evidence, and we fail to reject the null hypothesis.

[In practice, documentation of methods should answer the questions “what” and “why”. That is, “what” is the method, and “why” are we using/studying it? As this simulation experiment is a contrived example, we omit the “why” here.]

Function

#> function(X, y, cols = c("X1", "X2")) {
#>   lm_fit <- lm(y ~ X)
#>   pvals <- summary(lm_fit)$coefficients[cols, "Pr(>|t|)"] %>%
#>     setNames(paste(names(.), "p-value"))
#>   return(pvals)
#> }
#> <bytecode: 0x56031bdb1908>

Input Parameters

#> list()

Evaluation

Rejection Prob. (alpha = 0.1)

We define the rejection probability as the proportion of repetitions in the simulation experiment that result in a p-value \(\leq \alpha\). Here, we choose to set the significance level \(\alpha = 0.1\).

[In practice, documentation of evaluation metrics should answer the questions “what” and “why”. That is, “what” is the metric, and “why” are we using/studying it? As this simulation experiment is a contrived example, we omit the “why” here.]

Base

Function

#> function(fit_results, alpha = 0.05) {
#>   group_vars <- c(".dgp_name", ".method_name")
#>   eval_out <- fit_results %>%
#>     dplyr::group_by(across({{group_vars}})) %>%
#>     dplyr::summarise(
#>       `X1 Reject Prob.` = mean(`X1 p-value` < alpha),
#>       `X2 Reject Prob.` = mean(`X2 p-value` < alpha)
#>     )
#>   return(eval_out)
#> }

Input Parameters

#> $alpha
#> [1] 0.1
Linear Gaussian DGP - Varying beta

Function

#> function(fit_results, vary_params = NULL, alpha = 0.05) {
#>   group_vars <- c(".dgp_name", ".method_name", vary_params)
#>   eval_out <- fit_results %>%
#>     dplyr::group_by(across({{group_vars}})) %>%
#>     dplyr::summarise(
#>       `X1 Reject Prob.` = mean(`X1 p-value` < alpha),
#>       `X2 Reject Prob.` = mean(`X2 p-value` < alpha)
#>     )
#>   return(eval_out)
#> }
#> <bytecode: 0x56031ac28060>

Input Parameters

#> $alpha
#> [1] 0.1
Linear Gaussian DGP - Varying rho

Function

#> function(fit_results, vary_params = NULL, alpha = 0.05) {
#>   group_vars <- c(".dgp_name", ".method_name", vary_params)
#>   eval_out <- fit_results %>%
#>     dplyr::group_by(across({{group_vars}})) %>%
#>     dplyr::summarise(
#>       `X1 Reject Prob.` = mean(`X1 p-value` < alpha),
#>       `X2 Reject Prob.` = mean(`X2 p-value` < alpha)
#>     )
#>   return(eval_out)
#> }
#> <bytecode: 0x56031bbcecc8>

Input Parameters

#> $alpha
#> [1] 0.1
Linear Gaussian DGP - Varying sigma

Function

#> function(fit_results, vary_params = NULL, alpha = 0.05) {
#>   group_vars <- c(".dgp_name", ".method_name", vary_params)
#>   eval_out <- fit_results %>%
#>     dplyr::group_by(across({{group_vars}})) %>%
#>     dplyr::summarise(
#>       `X1 Reject Prob.` = mean(`X1 p-value` < alpha),
#>       `X2 Reject Prob.` = mean(`X2 p-value` < alpha)
#>     )
#>   return(eval_out)
#> }

Input Parameters

#> $alpha
#> [1] 0.1

Visualizations

Power

To examine the power of the test, we plot the rejection probability as a function of \(\alpha\), that is, \(\mathbb{P}(\text{p-value} \leq \alpha)\) vs. \(\alpha\). If the coefficient is non-zero in the underlying DGP, then a larger AUC would indicate better performance in terms of the power. We will primarily focus on plotting the power of the first coefficient \(\beta_1\).

[In practice, documentation of the plotters should answer the questions “what” and “why”. That is, “what” is the plot, and “why” are we using/studying it? As this simulation experiment is a contrived example, we omit the “why” here.]

Base

Function

#> function(fit_results, col = "X1") {
#>   plt <- ggplot2::ggplot(fit_results) +
#>     ggplot2::aes(x = .data[[paste(col, "p-value")]],
#>                  color = as.factor(.method_name)) +
#>     ggplot2::geom_abline(slope = 1, intercept = 0,
#>                          color = "darkgray", linetype = "solid", size = 1) +
#>     ggplot2::stat_ecdf(size = 1) +
#>     ggplot2::scale_x_continuous(limits = c(0, 1)) +
#>     ggplot2::labs(x = "t", y = "P( p-value \u2264 t )",
#>                   linetype = "", color = "Method")
#>   return(plt)
#> }

Input Parameters

#> list()
Linear Gaussian DGP - Varying beta

Function

#> function(fit_results, vary_params = NULL, col = "X1") {
#> 
#>   if (!is.null(vary_params)) {
#>     # deal with the case when we vary across a parameter that is vector-valued
#>     if (is.list(fit_results[[vary_params]])) {
#>       fit_results[[vary_params]] <- list_col_to_chr(fit_results[[vary_params]],
#>                                                     name = vary_params,
#>                                                     verbatim = TRUE)
#>     }
#>   }
#> 
#>   plt <- ggplot2::ggplot(fit_results) +
#>     ggplot2::aes(x = .data[[paste(col, "p-value")]],
#>                  color = as.factor(.method_name)) +
#>     ggplot2::geom_abline(slope = 1, intercept = 0,
#>                          color = "darkgray", linetype = "solid", size = 1) +
#>     ggplot2::stat_ecdf(size = 1) +
#>     ggplot2::scale_x_continuous(limits = c(0, 1)) +
#>     ggplot2::labs(x = "t", y = "P( p-value \u2264 t )",
#>                   linetype = "", color = "Method")
#>   if (!is.null(vary_params)) {
#>     plt <- plt + ggplot2::facet_wrap(~ .data[[vary_params]])
#>   }
#>   return(plt)
#> }
#> <bytecode: 0x560315970b18>

Input Parameters

#> list()
Linear Gaussian DGP - Varying rho

Function

#> function(fit_results, vary_params = NULL, col = "X1") {
#> 
#>   if (!is.null(vary_params)) {
#>     # deal with the case when we vary across a parameter that is vector-valued
#>     if (is.list(fit_results[[vary_params]])) {
#>       fit_results[[vary_params]] <- list_col_to_chr(fit_results[[vary_params]],
#>                                                     name = vary_params,
#>                                                     verbatim = TRUE)
#>     }
#>   }
#> 
#>   plt <- ggplot2::ggplot(fit_results) +
#>     ggplot2::aes(x = .data[[paste(col, "p-value")]],
#>                  color = as.factor(.method_name)) +
#>     ggplot2::geom_abline(slope = 1, intercept = 0,
#>                          color = "darkgray", linetype = "solid", size = 1) +
#>     ggplot2::stat_ecdf(size = 1) +
#>     ggplot2::scale_x_continuous(limits = c(0, 1)) +
#>     ggplot2::labs(x = "t", y = "P( p-value \u2264 t )",
#>                   linetype = "", color = "Method")
#>   if (!is.null(vary_params)) {
#>     plt <- plt + ggplot2::facet_wrap(~ .data[[vary_params]])
#>   }
#>   return(plt)
#> }
#> <bytecode: 0x56031505b8d8>

Input Parameters

#> list()
Linear Gaussian DGP - Varying sigma

Function

#> function(fit_results, vary_params = NULL, col = "X1") {
#> 
#>   if (!is.null(vary_params)) {
#>     # deal with the case when we vary across a parameter that is vector-valued
#>     if (is.list(fit_results[[vary_params]])) {
#>       fit_results[[vary_params]] <- list_col_to_chr(fit_results[[vary_params]],
#>                                                     name = vary_params,
#>                                                     verbatim = TRUE)
#>     }
#>   }
#> 
#>   plt <- ggplot2::ggplot(fit_results) +
#>     ggplot2::aes(x = .data[[paste(col, "p-value")]],
#>                  color = as.factor(.method_name)) +
#>     ggplot2::geom_abline(slope = 1, intercept = 0,
#>                          color = "darkgray", linetype = "solid", size = 1) +
#>     ggplot2::stat_ecdf(size = 1) +
#>     ggplot2::scale_x_continuous(limits = c(0, 1)) +
#>     ggplot2::labs(x = "t", y = "P( p-value \u2264 t )",
#>                   linetype = "", color = "Method")
#>   if (!is.null(vary_params)) {
#>     plt <- plt + ggplot2::facet_wrap(~ .data[[vary_params]])
#>   }
#>   return(plt)
#> }
#> <bytecode: 0x560314b76628>

Input Parameters

#> list()

Rejection Prob. (alpha = 0.1) Plot

We plot the rejection probability for \(\beta_1\) across varying parameters of the DGP to understand how characteristics of the DGP affect the test.We define the rejection probability as the proportion of repetitions in the simulation experiment that result in a p-value \(\leq \alpha\). Here, we choose to set the significance level \(\alpha = 0.1\).

[In practice, documentation of the plotters should answer the questions “what” and “why”. That is, “what” is the plot, and “why” are we using/studying it? As this simulation experiment is a contrived example, we omit the “why” here.]

Linear Gaussian DGP - Varying beta

Function

#> function(eval_results, vary_params = NULL,
#>                                  alpha = 0.05) {
#>   eval_results <- eval_results$`Rejection Prob. (alpha = 0.1)`
#>   if (is.list(eval_results[[vary_params]])) {
#>     # deal with the case when we vary across a parameter that is vector-valued
#>     eval_results[[vary_params]] <- list_col_to_chr(eval_results[[vary_params]],
#>                                                    name = vary_params,
#>                                                    verbatim = TRUE)
#>   }
#>   plt <- ggplot2::ggplot(eval_results) +
#>     ggplot2::aes(x = .data[[vary_params]], y = `X1 Reject Prob.`,
#>                  color = as.factor(.method_name),
#>                  fill = as.factor(.method_name)) +
#>     ggplot2::labs(x = vary_params,
#>                   y = sprintf("Rejection Probability (alpha = %s)", alpha),
#>                   color = "Method", fill = "Method") +
#>     ggplot2::scale_y_continuous(limits = c(0, 1))
#>   if (is.numeric(eval_results[[vary_params]])) {
#>     plt <- plt +
#>       ggplot2::geom_line() +
#>       ggplot2::geom_point(size = 2)
#>   } else {
#>     plt <- plt +
#>       ggplot2::geom_bar(stat = "identity")
#>   }
#>   return(plotly::ggplotly(plt))
#> }
#> <bytecode: 0x5603158bf550>

Input Parameters

#> $alpha
#> [1] 0.1
Linear Gaussian DGP - Varying rho

Function

#> function(eval_results, vary_params = NULL,
#>                                  alpha = 0.05) {
#>   eval_results <- eval_results$`Rejection Prob. (alpha = 0.1)`
#>   if (is.list(eval_results[[vary_params]])) {
#>     # deal with the case when we vary across a parameter that is vector-valued
#>     eval_results[[vary_params]] <- list_col_to_chr(eval_results[[vary_params]],
#>                                                    name = vary_params,
#>                                                    verbatim = TRUE)
#>   }
#>   plt <- ggplot2::ggplot(eval_results) +
#>     ggplot2::aes(x = .data[[vary_params]], y = `X1 Reject Prob.`,
#>                  color = as.factor(.method_name),
#>                  fill = as.factor(.method_name)) +
#>     ggplot2::labs(x = vary_params,
#>                   y = sprintf("Rejection Probability (alpha = %s)", alpha),
#>                   color = "Method", fill = "Method") +
#>     ggplot2::scale_y_continuous(limits = c(0, 1))
#>   if (is.numeric(eval_results[[vary_params]])) {
#>     plt <- plt +
#>       ggplot2::geom_line() +
#>       ggplot2::geom_point(size = 2)
#>   } else {
#>     plt <- plt +
#>       ggplot2::geom_bar(stat = "identity")
#>   }
#>   return(plotly::ggplotly(plt))
#> }
#> <bytecode: 0x560314f28790>

Input Parameters

#> $alpha
#> [1] 0.1

Base Linear Regression Experiment

Rejection Prob. (alpha = 0.1)

Power

Linear Gaussian DGP

Varying beta

Rejection Prob. (alpha = 0.1)

Power

Rejection Prob. (alpha = 0.1) Plot

Parameter Values

#> $dgp
#> $dgp$`Linear Gaussian DGP`
#> $dgp$`Linear Gaussian DGP`$beta
#> $dgp$`Linear Gaussian DGP`$beta[[1]]
#> [1] 1 0
#> 
#> $dgp$`Linear Gaussian DGP`$beta[[2]]
#> [1] 1.0 0.5
#> 
#> $dgp$`Linear Gaussian DGP`$beta[[3]]
#> [1] 1 1
#> 
#> $dgp$`Linear Gaussian DGP`$beta[[4]]
#> [1] 1.0 1.5
#> 
#> $dgp$`Linear Gaussian DGP`$beta[[5]]
#> [1] 1 2
#> 
#> 
#> 
#> 
#> $method
#> list()

Varying rho

Rejection Prob. (alpha = 0.1)

Power

Rejection Prob. (alpha = 0.1) Plot

Parameter Values

#> $dgp
#> $dgp$`Linear Gaussian DGP`
#> $dgp$`Linear Gaussian DGP`$rho
#> [1] 0.0 0.2 0.5 0.9
#> 
#> 
#> 
#> $method
#> list()

Varying sigma

Rejection Prob. (alpha = 0.1)

Power

Parameter Values

#> $dgp
#> $dgp$`Linear Gaussian DGP`
#> $dgp$`Linear Gaussian DGP`$sigma
#> [1] 1 2 4 8
#> 
#> 
#> 
#> $method
#> list()
---
title: "`r params$sim_name`"
author: "`r params$author`"
date: "`r format(Sys.time(), '%B %d, %Y')`"
header-includes:
    - \usepackage{float}
    - \usepackage{amsmath}
    - \usepackage{gensymb}
output:
  vthemes::vmodern:
    thumbnails: false
css: css/simchef.css
params:
  author: 
    label: "Author:"
    value: ""
  sim_name:
    label: "Simulation Experiment Name:"
    value: ""
  sim_path:
    label: "Path to Simulation Experiment Folder:"
    value: ""
  eval_order:
    label: "Order of Evaluators:"
    value: NULL
  viz_order:
    label: "Order of Visualizers:"
    value: NULL
  verbose:
    label: "Verbose Level:"
    value: 2
---

<script src="js/simchefNavClass.js"></script>

```{r setup, include=FALSE}
options(width = 10000)
knitr::opts_chunk$set(
  echo = FALSE,
  warning = FALSE,
  message = FALSE,
  cache = FALSE,
  fig.align = "center",
  fig.pos = "H",
  fig.height = 12,
  fig.width = 10
)

options(knitr.kable.NA = 'NA',
        dplyr.summarise.inform = FALSE)

# scrollable text output
local({
  hook_output <- knitr::knit_hooks$get('output')
  knitr::knit_hooks$set(output = function(x, options) {
    if (!is.null(options$max.height)) options$attr.output <- c(
      options$attr.output,
      sprintf('style="max-height: %s;"', options$max.height)
    )
    hook_output(x, options)
  })
})

chunk_idx <- 1
doc_dir <- file.path(params$sim_path, "docs")
```

```{r helper-funs}

#' Get order of objects to display
#'
#' @param obj_names Vector of all object names that need to be displayed.
#' @param obj_order Vector of object names in the desired appearance order.
#' @return Vector of object names in the order in which they will be displayed.
getObjOrder <- function(obj_names, obj_order = NULL) {
  if (is.null(obj_order)) {
    return(obj_names)
  } else {
    return(intersect(obj_order, obj_names))
  }
}

#' Get all experiments under a given directory name
#'
#' @param dir_name name of directory
#' @return list of named experiments
getDescendants <- function(dir_name) {
  experiments <- list()
  for (d in list.dirs(dir_name)) {
    if (file.exists(file.path(d, "experiment.rds"))) {
      if (identical(d, params$sim_path)) {
        exp_name <- "Base"
      } else {
        exp_name <- stringr::str_replace_all(
          stringr::str_remove(d, paste0(params$sim_path, "/")),
          "/", " - "
        )
      }
      experiments[[exp_name]] <- readRDS(file.path(d, "experiment.rds"))
    }
  }
  return(experiments)
}

#' Check if experiment exists
#'
#' @param dir_name name of directory or vector thereof
#' @param recursive logical; if TRUE, checks if experiment exists under the
#'   given directory(s); if FALSE, checks if any experiment exists under the
#'   directory(s) and its descendants
#' @return TRUE if experiment exists and FALSE otherwise
experimentExists <- function(dir_name, recursive = FALSE) {
  res <- purrr::map_lgl(dir_name,
                        function(d) {
                          if (!recursive) {
                            exp_fname <- file.path(d, "experiment.rds")
                            return(file.exists(exp_fname))
                          } else {
                            descendants <- getDescendants(d)
                            return(length(descendants) > 0)
                          }
                        })
  return(any(res))
}

#' Displays content for specified part of recipe
#'
#' @param field_name part of recipe to show; must be one of "dgp", "method",
#'   "evaluator", or "visualizer"
#' @return content for recipe
showRecipePart <- function(field_name = c("dgp", "method",
                                          "evaluator", "visualizer")) {

  field_name <- match.arg(field_name)
  func_name <- dplyr::case_when(field_name == "evaluator" ~ "eval",
                                field_name == "visualizer" ~ "viz",
                                TRUE ~ field_name)
  descendants <- getDescendants(dir_name = params$sim_path)
  objs <- purrr::map(descendants, ~.x[[paste0("get_", field_name, "s")]]())
  obj_names <- unique(purrr::reduce(sapply(objs, names), c))

  obj_header <- "<p style='font-weight: bold; font-size: 20px'> %s </p>"
  invis_header <- "\n\n### %s {.tabset .tabset-pills .tabset-recipe .tabset-circle}\n\n"
  showtype_header <- "\n\n#### %s {.tabset .tabset-pills}\n\n"
  exp_header <- "\n\n##### %s \n\n"

  if (all(sapply(objs, length) == 0)) {
    return(cat("N/A"))
  }

  for (idx in 1:length(obj_names)) {
    cat(sprintf(invis_header, ""))
    obj_name <- obj_names[idx]

    cat("<div class='panel panel-default padded-panel'>")
    cat(sprintf(obj_header, obj_name))

    cat(sprintf(showtype_header, fontawesome::fa("readme", fill = "white")))
    pasteMd(file.path(doc_dir, paste0(field_name, "s"),
                      paste0(obj_name, ".md")))

    cat(sprintf(showtype_header, fontawesome::fa("code", fill = "white")))
    keep_objs <- purrr::map(objs, obj_name)
    keep_objs[sapply(keep_objs, is.null)] <- NULL
    if (all(purrr::map_lgl(keep_objs,
                           ~isTRUE(check_equal(.x, keep_objs[[1]]))))) {
      obj <- keep_objs[[1]]
      cat("<b>Function</b>")
      vthemes::subchunkify(obj[[paste0(func_name, "_fun")]],
                  chunk_idx, other_args = "max.height='200px'")
      chunk_idx <<- chunk_idx + 1
      cat("<b>Input Parameters</b>")
      vthemes::subchunkify(obj[[paste0(func_name, "_params")]],
                  chunk_idx, other_args = "max.height='200px'")
      chunk_idx <<- chunk_idx + 1
    } else {
      for (exp in names(objs)) {
        obj <- objs[[exp]][[obj_name]]
        if (is.null(obj)) {
          next
        }
        cat(sprintf(exp_header, exp))
        cat("<b>Function</b>")
        vthemes::subchunkify(obj[[paste0(func_name, "_fun")]],
                    chunk_idx, other_args = "max.height='200px'")
        chunk_idx <<- chunk_idx + 1
        cat("<b>Input Parameters</b>")
        vthemes::subchunkify(obj[[paste0(func_name, "_params")]],
                    chunk_idx, other_args = "max.height='200px'")
        chunk_idx <<- chunk_idx + 1
      }
    }
    cat("</div>")
  }
}

#' Reads in file if it exists and returns NULL if the file does not exist
#'
#' @param filename name of .rds file to try reading in
#' @return output of filename.rds if the file exists and NULL otherwise
getResults <- function(filename) {
  if (file.exists(filename)) {
    results <- readRDS(filename)
  } else {
    results <- NULL
  }
  return(results)
}

#' Displays output (both from evaluate() and visualize()) from saved results under
#' a specified directory
#'
#' @param dir_name name of directory
#' @param depth integer; depth of directory from parent/base experiment's folder
#' @param base logical; whether or not this is a base experiment
#' @param show_header logical; whether or not to show section header
#' @param verbose integer; 0 = no messages; 1 = print out directory name only;
#'   2 = print out directory name and name of evaluators/visualizers
#' @return content results from evaluate() and visualize() from the experiment
showResults <- function(dir_name, depth, base = FALSE, show_header = TRUE,
                        verbose = 1) {
  if (verbose >= 1) {
    message(rep("*", depth), basename(dir_name))
  }

  if (depth == 1) {
    header_template <- "\n\n%s %s {.tabset .tabset-pills .tabset-vmodern}\n\n"
  } else {
    if (base | !experimentExists(dir_name)) {
      header_template <- "\n\n%s %s {.tabset .tabset-pills}"
    } else {
      header_template <- "\n\n%s %s {.tabset .tabset-pills .tabset-circle}"
    }
  }

  if (show_header) {
    cat(sprintf(header_template,
                paste(rep("#", depth), collapse = ""),
                basename(dir_name)))
  }

  if (base) {
    cat(paste0("\n\n",
               paste(rep("#", depth + 1), collapse = ""),
               " Base - ", basename(dir_name),
               " {.tabset .tabset-pills .tabset-circle}\n\n"))
    depth <- depth + 1
  }

  showtype_template <- paste0(
    "\n\n", paste(rep("#", depth + 1), collapse = ""), " %s\n\n"
  )
  figname_template <- "<h3 style='font-weight: bold'> %s </h3>"
  invisible_header <- paste0(
    "\n\n", paste(rep("#", depth + 2), collapse = ""),
    " {.tabset .tabset-pills}\n\n"
  )
  plt_template <- paste0(
    "\n\n", paste(rep("#", depth + 3), collapse = ""), " %s\n\n"
  )

  exp_fname <- file.path(dir_name, "experiment.rds")
  # fit_fname <- file.path(dir_name, "fit_results.rds")
  eval_fname <- file.path(dir_name, "eval_results.rds")
  viz_fname <- file.path(dir_name, "viz_results.rds")

  exp <- getResults(exp_fname)
  # fit_results <- getResults(fit_fname)
  eval_results <- getResults(eval_fname)
  viz_results <- getResults(viz_fname)

  if (!is.null(eval_results)) {
    cat(sprintf(showtype_template, fontawesome::fa("table", fill = "white")))
    eval_names <- getObjOrder(names(eval_results), params$eval_order)
    for (eval_name in eval_names) {
      if (verbose >= 2) {
        message(rep(" ", depth + 1), eval_name)
      }
      evaluator <- exp$get_evaluators()[[eval_name]]
      if (evaluator$doc_show) {
        cat(sprintf(figname_template, eval_name))
        do.call(vthemes::pretty_DT,
                c(list(eval_results[[eval_name]]), evaluator$doc_options)) %>%
          vthemes::subchunkify(i = chunk_idx)
        chunk_idx <<- chunk_idx + 1
      }
    }
  }

  if (!is.null(viz_results)) {
    cat(sprintf(showtype_template,
                fontawesome::fa("chart-bar", fill = "white")))
    viz_names <- getObjOrder(names(viz_results), params$viz_order)
    for (viz_name in viz_names) {
      if (verbose >= 2) {
        message(rep(" ", depth + 1), viz_name)
      }
      visualizer <- exp$get_visualizers()[[viz_name]]
      if (visualizer$doc_show) {
        cat(invisible_header)
        cat(sprintf(figname_template, viz_name))
        plts <- viz_results[[viz_name]]
        if (!inherits(plts, "list")) {
          plts <- list(plt = plts)
        }
        if (is.null(names(plts))) {
          names(plts) <- 1:length(plts)
        }
        for (plt_name in names(plts)) {
          if (length(plts) != 1) {
            cat(sprintf(plt_template, plt_name))
          }
          plt <- plts[[plt_name]]
          if (inherits(plt, "plotly") | inherits(plt, "gg") | inherits(plt, "ggplot")) {
            add_class <- c("panel panel-default padded-panel")
          } else {
            add_class <- NULL
          }
          vthemes::subchunkify(plt, i = chunk_idx,
                      fig_height = visualizer$doc_options$height,
                      fig_width = visualizer$doc_options$width,
                      other_args = "out.width = '100%'",
                      add_class = add_class)
          chunk_idx <<- chunk_idx + 1
        }
      }
    }
  }

  if (!is.null(exp)) {
    if ((length(exp$get_vary_across()$dgp) != 0) |
        (length(exp$get_vary_across()$method) != 0)) {
      cat(sprintf(showtype_template, fontawesome::fa("code", fill = "white")))
      cat("<b>Parameter Values</b>")
      vthemes::subchunkify(exp$get_vary_across(),
                  chunk_idx, other_args = "max.height='200px'")
      chunk_idx <<- chunk_idx + 1
    }
  }
}

#' Displays output of experiment for all of its (saved) descendants
#'
#' @param dir_name name of parent experiment directory
#' @param depth placeholder for recursion; should not be messed with
#' @param ... other arguments to pass into showResults()
showDescendantResults <- function(dir_name, depth = 1, ...) {
  children <- list.dirs(dir_name, recursive = FALSE)
  if (length(children) == 0) {
    return()
  }
  for (child_idx in 1:length(children)) {
    child <- children[child_idx]
    if (!experimentExists(child, recursive = TRUE)) {
      next
    }
    if (experimentExists(child, recursive = FALSE) &
        (experimentExists(list.dirs(child, recursive = TRUE)[-1]) |
         (depth == 1))) {
      base <- TRUE
    } else {
      base <- FALSE
    }
    showResults(child, depth, base = base, ...)
    showDescendantResults(child, depth + 1, ...)
  }
}


```

# Simulation Experiment Recipe {.tabset .tabset-vmodern}

## Objectives {.panel .panel-default .padded-panel}

```{r objectives, results = "asis"}
pasteMd(file.path(doc_dir, "objectives.md"))
```

## Data Generation

```{r dgps, results = "asis"}
showRecipePart(field_name = "dgp")
```

## Methods and Evaluation

### Methods

```{r methods, results = "asis"}
showRecipePart(field_name = "method")
```

### Evaluation

```{r evaluators, results = "asis"}
showRecipePart(field_name = "evaluator")
```

## Visualizations

```{r visualizers, results = "asis"}
showRecipePart(field_name = "visualizer")
```



```{r res, results = "asis"}

# show results
if (experimentExists(params$sim_path)) {
  cat(sprintf("\n\n# Base %s \n\n", params$sim_name))
  cat("\n\n## {.tabset .tabset-pills .tabset-circle}\n\n")
  message(sprintf("Creating R Markdown report for %s...", params$sim_name))
  showResults(params$sim_path, depth = 2, base = FALSE, show_header = FALSE,
              verbose = 0)
}

showDescendantResults(params$sim_path, verbose = params$verbose)

```
